package zw_201_300.zw_283_移动零;

class Solution {

//    public static void main(String[] args) {
//        int[] nums = {0,1,0,3,12};
//        int[] n = moveZeroes(nums);
//        for (int i = 0; i < n.length; i++) {
//            System.out.println(n[i]);
//        }
//
//    }

//    public static int[] moveZeroes(int[] nums) {
//        int index = 0;
//        for (int num : nums) {
//            if (num != 0) {
//                nums[index++] = num;
//            }
//        }
//        for (int i = index; i < nums.length; i++) {
//            nums[i] = 0;
//        }
//        return nums;
//    }


    /**
     * 双指针
     * 思路及解法
     * 使用双指针，左指针指向当前已经处理好的序列的尾部，右指针指向待处理序列的头部。
     * 右指针不断向右移动，每次右指针指向非零数，则将左右指针对应的数交换，同时左指针右移。
     * 注意到以下性质：
     * 左指针左边均为非零数；
     * 右指针左边直到左指针处均为零。
     * 因此每次交换，都是将左指针的零与右指针的非零数交换，且非零数的相对顺序并未改变。
     * 时间复杂度：O(n)，其中 n 为序列长度。每个位置至多被遍历两次。
     * 空间复杂度：O(1)。只需要常数的空间存放若干变量。
     *
     * @param nums
     */
    public void moveZeroes(int[] nums) {
        int n = nums.length, left = 0, right = 0;
        while (right < n) {
            if (nums[right] != 0) {
                swap(nums, left, right);
                left++;
            }
            right++;
        }
    }

    public void swap(int[] nums, int left, int right) {
        int temp = nums[left];
        nums[left] = nums[right];
        nums[right] = temp;
    }


    /**
     * 覆盖，后面补0
     *
     * @param nums
     */
//    public void moveZeroes(int[] nums) {
//        int index = 0;
//        for (int num : nums) {
//            if (num != 0) {
//                nums[index++] = num;
//            }
//        }
//        for (int i = index; i < nums.length; i++) {
//            nums[i] = 0;
//        }
//    }
}
